/*
 * Copyright (c) 2012 Centre Tecnologic de Telecomunicacions de Catalunya (CTTC)
 *
 * SPDX-License-Identifier: GPL-2.0-only
 *
 * Author: Marco Miozzo  <marco.miozzo@cttc.es>
 */

#ifndef LTE_HARQ_PHY_MODULE_H
#define LTE_HARQ_PHY_MODULE_H

#include "ns3/assert.h"
#include "ns3/log.h"
#include "ns3/simple-ref-count.h"

#include <map>
#include <math.h>
#include <vector>

namespace ns3
{

/// HarqProcessInfoElement_t structure
struct HarqProcessInfoElement_t
{
    double m_mi;         ///< Mutual information
    uint8_t m_rv;        ///< Redundancy version
    uint16_t m_infoBits; ///< info bits
    uint16_t m_codeBits; ///< code bits
};

typedef std::vector<HarqProcessInfoElement_t>
    HarqProcessInfoList_t; ///< HarqProcessInfoList_t typedef

/**
 * @ingroup lte
 * @brief The LteHarqPhy class implements the HARQ functionalities related to PHY layer
 *(i.e., decodification buffers for incremental redundancy management)
 *
 */
class LteHarqPhy : public SimpleRefCount<LteHarqPhy>
{
  public:
    LteHarqPhy();
    ~LteHarqPhy();

    /**
     * @brief Subframe Indication function
     * @param frameNo the frame number
     * @param subframeNo the subframe number
     */
    void SubframeIndication(uint32_t frameNo, uint32_t subframeNo);

    /**
     * @brief Return the cumulated MI of the HARQ procId in case of retransmissions
     * for DL (asynchronous)
     * @param harqProcId the HARQ proc id
     * @param layer layer no. (for MIMO spatial multiplexing)
     * @return the MI accumulated
     */
    double GetAccumulatedMiDl(uint8_t harqProcId, uint8_t layer);

    /**
     * @brief Return the info of the HARQ procId in case of retransmissions
     * for DL (asynchronous)
     * @param harqProcId the HARQ proc id
     * @param layer layer no. (for MIMO spatial multiplexing)
     * @return the vector of the info related to HARQ proc Id
     */
    HarqProcessInfoList_t GetHarqProcessInfoDl(uint8_t harqProcId, uint8_t layer);

    /**
     * @brief Return the cumulated MI of the HARQ procId in case of retransmissions
     * for UL (synchronous)
     * @param rnti the RNTI of the transmitter
     * @return the MI accumulated
     */
    double GetAccumulatedMiUl(uint16_t rnti);

    /**
     * @brief Return the info of the HARQ procId in case of retransmissions
     * for UL (asynchronous)
     * @param rnti the RNTI of the transmitter
     * @param harqProcId the HARQ proc id
     * @return the vector of the info related to HARQ proc Id
     */
    HarqProcessInfoList_t GetHarqProcessInfoUl(uint16_t rnti, uint8_t harqProcId);

    /**
     * @brief Update the Info associated to the decodification of an HARQ process
     * for DL (asynchronous)
     * @param id the HARQ proc id
     * @param layer layer no. (for MIMO spatial multiplexing)
     * @param mi the new MI
     * @param infoBytes the no. of bytes of info
     * @param codeBytes the total no. of bytes txed
     */
    void UpdateDlHarqProcessStatus(uint8_t id,
                                   uint8_t layer,
                                   double mi,
                                   uint16_t infoBytes,
                                   uint16_t codeBytes);

    /**
     * @brief Reset the info associated to the decodification of an HARQ process
     * for DL (asynchronous)
     * @param id the HARQ proc id
     */
    void ResetDlHarqProcessStatus(uint8_t id);

    /**
     * @brief Update the MI value associated to the decodification of an HARQ process
     * for DL (asynchronous)
     * @param rnti the RNTI of the transmitter
     * @param mi the new MI
     * @param infoBytes the no. of bytes of info
     * @param codeBytes the total no. of bytes txed
     */
    void UpdateUlHarqProcessStatus(uint16_t rnti,
                                   double mi,
                                   uint16_t infoBytes,
                                   uint16_t codeBytes);

    /**
     * @brief Reset  the info associated to the decodification of an HARQ process
     * for DL (asynchronous)
     * @param rnti the RNTI of the transmitter
     * @param id the HARQ proc id
     */
    void ResetUlHarqProcessStatus(uint16_t rnti, uint8_t id);

    /**
     * @brief Clear the downlink HARQ buffer
     *
     * @param rnti the RNTI of the UE
     */
    void ClearDlHarqBuffer(uint16_t rnti);

  private:
    std::vector<std::vector<HarqProcessInfoList_t>>
        m_miDlHarqProcessesInfoMap; ///< MI DL HARQ processes info map
    std::map<uint16_t, std::vector<HarqProcessInfoList_t>>
        m_miUlHarqProcessesInfoMap; ///< MI UL HARQ processes info map
};

} // namespace ns3

#endif /* LTE_HARQ_PHY_MODULE_H */
